home *** CD-ROM | disk | FTP | other *** search
/ TeX 1995 July / TeX CD-ROM July 1995 (Disc 1)(Walnut Creek)(1995).ISO / macros / latex209 / contrib / misc / catmac-doc.tex (.txt) < prev    next >
LaTeX Document  |  1990-04-28  |  18KB  |  501 lines

  1. %Please reply to inhb@mcgillb.bitnet.
  2. %For this semester only, to barr@linc.cis.upenn.edu.
  3. %Michael Barr
  4. %Math Dept
  5. %McGill Univ.
  6. %805 Sherbrooke St W
  7. %Montreal, QC
  8. %Canada H3P 1S4
  9. \documentstyle[12pt,catmac]{article}
  10. \textheight=9in \topmargin=0pt\headheight=0pt\headsep=0pt
  11. \textwidth=6.5in \oddsidemargin=0pt
  12. \begin{document}
  13. {\catcode`\ =13\global\let =\ \catcode`\^^M=13
  14. \gdef^^M{\par\noindent}}
  15. \def\verbatim{\tt
  16. \catcode`\^^M=13
  17. \catcode`\ =13
  18. \catcode`\\=12
  19. \catcode`\{=12
  20. \catcode`\}=12
  21. \catcode`\_=12
  22. \catcode`\^=12
  23. \catcode`\&=12
  24. \catcode`\~=12
  25. \catcode`\#=12
  26. \catcode`\%=12
  27. \catcode`\$=12
  28. \catcode`|=0
  29. All commercial rights reserved.  May be freely distributed
  30. and used with the following exceptions:\\
  31. 1. No commercial use without explicit permission.\\
  32. 2. It may not be used by any employee of a telephone company.\\
  33. 3. It may not be distributed without this notice.
  34. \def\\{{\tt \char 92}}
  35. \section*{The catmac macros}
  36. The file catmac includes a number of macros for drawing commutative
  37. diagrams in special shapes as well as more flexible macros to paste
  38. together more complicated diagrams.
  39. In general, there are two kinds of macros.  The first kind has the
  40. following form
  41. {\tt \\shapename[shape\_parameters]} and is intended to be used in the
  42. following way:
  43. {\verbatim
  44. \begin{center}
  45. \resetparms
  46. \shapename[shape_parameters]
  47. \end{center}
  48. |egroup
  49. This will create a shape with the given parameters.  It operates by
  50. building its own picture environment.
  51. The second kind of procedure must be used inside a picture environment
  52. and allows the user to put a diagram of a given size and shape at a
  53. given place.  The usual way in which this is used is:
  54. {\verbatim
  55. \begin{center}
  56. %code to be explained later
  57. \begin{picture}(xext,yext)
  58. \putshape1(x1,y1)[shape_parameters1]
  59. |dots
  60. \putshape(xn,yn)[shape_parametersn]
  61. \end{picture}
  62. \end{center}
  63. |egroup
  64. Here is an example of the first kind of macro.  I will have to explain a
  65. few details before giving examples of the second.  The code
  66. {\verbatim
  67. \begin{center}
  68. \resetparms
  69. \square[A`B`C`D;f`g`h`k]
  70. \end{center}
  71. |egroup
  72. produces the diagram
  73. \begin{center}
  74. \resetparms
  75. \square[A`B`C`D;f`g`h`k]
  76. \end{center}
  77. and the code
  78. {\verbatim
  79. \begin{center}
  80. \resetparms
  81. \Atriangle[A`B`C;f`g`h]
  82. \end{center}
  83. |egroup
  84. produces the diagram
  85. \begin{center}
  86. \resetparms
  87. \Atriangle[A`B`C;f`g`h]
  88. \end{center}
  89. The reason for the shape name will be explained later.
  90. If an arrow label extends especially high or low, the space
  91. automatically expands to match.  For example,
  92. {\verbatim
  93. \begin{center}
  94. \resetparms
  95. \square[A`B`C`D;\sum_{i=1}^{\infty}`g`h`\Psi^A_k]
  96. \end{center}
  97. |egroup
  98. will get the diagram
  99. \begin{center}
  100. \resetparms
  101. \square[A`B`C`D;\sum_{i=1}^{\infty}`g`h`\Psi^A_k]
  102. \end{center}
  103. and so on.
  104. Before going on, I will explain about the parameters.  Any shape will
  105. have one or two parameters for height and/or length as well as one
  106. parameter for each arrow in the diagram.  These are in addition to the
  107. parameters used in the actual procedures.  For example, the procedure
  108. {\verbatim
  109. \begin{center}
  110. \setsqparms[-1`0`2`-3;1000`700]
  111. \square[A`B`C`D;f`g`h`k]
  112. \end{center}
  113. |egroup
  114. gives the square:
  115. \begin{center}
  116. \setsqparms[-1`0`2`-3;1000`700]
  117. \square[A`B`C`D;f`g`h`k]
  118. \end{center}
  119. The meaning is as follows.  The first four parameters refer to four
  120. arrows in linguistic order (top, left, right, bottom).  A negative
  121. number gives a backward arrow, while a zero causes it to be omitted.  A
  122. parameter with an absolute value of 1 is ordinary, while 2 gives an
  123. arrow with a tail (monomorphism) and 3 a double headed arrow
  124. (epimorphism).  The last two parameters determine the width and height,
  125. respectively, in units of 0.01 em (an em is the width of an M, the
  126. widest letter in a font).
  127. Although {\tt \\resetparms} works with any of the shapes (and simply
  128. chooses
  129. the default values of 1 for all arrows and 500 for height and width),
  130. the remaining shapes use different names.  The various parameter setting
  131. procedures are
  132. {\verbatim
  133. \setsqparms[#1`#2`#3`#4;#5`#6]
  134. \settriparms[#1`#2`#3;#4]
  135. \settripairparms[#1`#2`#3`#4`#5;#6]
  136. \setrecparms[#1`#2]
  137. |egroup
  138. With the exception of the last, the parameters before the semicolon set
  139. the arrow types and the one or two after set lengths.  In the last one,
  140. both parameters are lengths.
  141. Here are the shapes.  We have already seen the shape {\tt \\square}.
  142. There are eight different triangles, all isoceles right triangles in
  143. different orientations.  The names all have the form {\tt
  144. \\xtriangle}, where x is the letter that most closely resembles the
  145. actual shape of the triangle.
  146. Here are the names, followed by a sample of each one:
  147. \begin{list}{}{}
  148. \item{\tt \\btriangle}\samepage
  149. \begin{center}
  150. \resetparms
  151. \btriangle[A`B`C;f`g`h]
  152. \end{center}
  153. \pagebreak[0]
  154. \item{\tt \\dtriangle}\samepage
  155. \begin{center}
  156. \resetparms
  157. \dtriangle[A`B`C;f`g`h]
  158. \end{center}
  159. \pagebreak[0]
  160. \item{\tt \\ptriangle}\samepage
  161. \begin{center}
  162. \resetparms
  163. \ptriangle[A`B`C;f`g`h]
  164. \end{center}
  165. \pagebreak[0]
  166. \item{\tt \\qtriangle}\samepage
  167. \begin{center}
  168. \resetparms
  169. \qtriangle[A`B`C;f`g`h]
  170. \end{center}
  171. \pagebreak[0]
  172. \item{\tt \\Atriangle}\samepage
  173. \begin{center}
  174. \resetparms
  175. \Atriangle[A`B`C;f`g`h]
  176. \end{center}
  177. \pagebreak[0]
  178. \item{\tt \\Vtriangle}\samepage
  179. \begin{center}
  180. \resetparms
  181. \Vtriangle[A`B`C;f`g`h]
  182. \end{center}
  183. \pagebreak[0]
  184. \item{\tt \\Ctriangle}\samepage
  185. \begin{center}
  186. \resetparms
  187. \Ctriangle[A`B`C;f`g`h]
  188. \end{center}
  189. \pagebreak[0]
  190. \item{\tt \\Dtriangle}\samepage
  191. \begin{center}
  192. \resetparms
  193. \Dtriangle[A`B`C;f`g`h]
  194. \end{center}
  195. \end{list}
  196. In addition, there are two special diagrams that come up often enough to
  197. be worth having a special macros for.
  198. \begin{list}{}{}
  199. \item{\tt \\Atrianglepair}\samepage
  200. \begin{center}
  201. \resetparms
  202. \Atrianglepair[A`B`C`D;f`g`h`k`l]
  203. \end{center}
  204. \item{\tt \\Vtrianglepair}\samepage
  205. \begin{center}
  206. \resetparms
  207. \Vtrianglepair[A`B`C`D;f`g`h`k`l]
  208. \end{center}
  209. \end{list}
  210. Finally, there is one special shape that is probably not used by very
  211. many mathematicians.  Still I have it and the user might as well share
  212. {\verbatim
  213. \recurse[A`B`R`C;s`f_0`f`t_0`t]
  214. |egroup
  215. \begin{center}
  216. \resetparms
  217. \recurse[A`B`R`C;s`f_0`f`t_0`t]
  218. \end{center}
  219. Notice what happens if the first parameter is empty:
  220. {\verbatim
  221. \recurse[`B`R`C;s`f_0`f`t_0`t]
  222. |egroup
  223. \begin{center}
  224. \resetparms
  225. \recurse[`B`R`C;s`f_0`f`t_0`t]
  226. \end{center}
  227. This takes care of the simple procedures.  The remaining procedures are
  228. there as building blocks for more complicated diagrams.
  229. These take the form
  230. \vskip0pt\noindent
  231. {\tt \\putshape(xpos,ypos)[shape\_parameters]}
  232. \vskip0pt\noindent
  233. where
  234. shape is one of the 11 shapes (excluding {\tt \\recurse}) described
  235. above
  236. and the parms are as described there.  The parameters {\tt  xpos} and
  237. {\tt  ypos} are the offsets from the lower left corner of the picture
  238. measured in umits of .01 em in the usual \LaTeX\ fashion.
  239. There is one
  240. more shape which is simply an arrow with an attached label.  This is
  241. called {\tt \\putmorphism} and it is used with the syntax
  242. \vskip0pt\noindent
  243. \\putmorphism(xpos,ypos)(run,rise)[node1`node2`label]\{dist\}\{type\}\{loc\}}
  244. \vskip0pt\noindent
  245. Here the parameters {\tt  xpos} and
  246. {\tt  ypos} are as above.  The slope is {\tt rise/run} where {\tt rise}
  247. and {\tt run} are two numbers that give the slope in accordance with the
  248. \LaTeX\ rules.  That is rise and run must have no common divisor and
  249. must not exceed 4 in absolute value.  In addition, these procedures are
  250. defined so that rise must not be positive and if it is zero, then run
  251. must be positive.  In other words, all arrows must go in the linguistic
  252. direction, downwards or to the right.  Arrows can be made to go in the
  253. reverse direction as explained below.
  254. The next three parameters are the node that appears first (in linguistic
  255. order), the one that appears second and the arrow label.  The next
  256. paramater is the distance in the sense of \LaTeX\ between the centers of
  257. the nodes.  This means it is the horizontal component of the distance,
  258. unless that is negative, in which case it is the vertical distance.  The
  259. second parameter is the code for the arrowtype, an integer between -3
  260. and 3 with the same meaning as explained above.  Using negative values
  261. of this parameter allows one to draw an arrow that goes upwards or left.
  262. Note, however, that node1 is always the node that is above or to the
  263. left of node2.  The last parameter should have the value a,b,l,r or m.
  264. The values a and b are used only for horizontal arrows and direct the
  265. arrow label to be placed above or below the arrow.  The values l and r
  266. are used for all other arrows and direct the label to be left or right
  267. of the arrow.  Finally, a vertical arrow only can be given the parameter
  268. m, in which case the arrow will be gapped and the label placed in the
  269. middle of the gap.
  270. In general, the simple macros are designed to be used as indicated
  271. either in the {\tt \\begin\{center\} \ldots \\end\{center\}} environment
  272. or in one of the others, usually
  273. {\verbatim
  274. \begin{equation}\begin{array}{c}
  275. |dots
  276. \end{array}\end{equation}|egroup
  277. The \\putshape macros must be used inside a picture environment that is
  278. normally placed inside a centering or similar environment.  The
  279. reference point for the positioning parameters is determined as the
  280. lower left corner of the smallest rectangle with sides parallel to the
  281. coordinate axes that includes the center points of all nodes on its
  282. border.  This rectangle will be degenerate in the case of a horizontal
  283. or vertical morphism.  The reference point may either be on or outside
  284. the actual figure.  For example, in the case of the btriangle, it is the
  285. center of the lower left node, while for a qtriangle it is outside the
  286. triangle itself being the fourth corner of the enclosing square.
  287. The macros are made so that they fit together well.  That is why all
  288. distances are from node centers to node centers.  If a vertex is part of
  289. two shapes, it is probably best not to repeat it, since it is
  290. conceivable that round-off errors will cause its two appearances to be
  291. slightly offset.  It can either be omitted or, since the width is used
  292. to determine the amount to shorten horizontal arrows, replaced by {\tt
  293. \\phantom} versions.  Both methods are illustrated in the example given
  294. at the end.
  295. One more thing has to be described.  when beginning picture mode, you
  296. have to allocate both a horizontal and vertical space.  the vertical
  297. space is crucial and, while the horizontal is not as critical, it still
  298. determines whether the figure is properly centered.  This partially
  299. automated as follows.  You begin by giving values to two variables,
  300. called {\tt \\xext} and {\tt\\yext} (for x-extent and y-extent, resp.).
  301. This is done by the declarations
  302. {\tt\\xext=xparm, \\yext=yparm}, where {\tt xparm} and {\tt yparm} are
  303. the sum of all the horizontal and vertical distances used in the
  304. procedures.  Of course, if two shapes are put side by side, the
  305. x-extents will be the sum plus the space between them if any, while the
  306. y-extent will be that of the larger one.  In figuring out these extents,
  307. ignore the heights and widths of the vertices and arrow labels.  So what
  308. these really measure is the distance between the two most distantly
  309. separated arrows in the diagram, or rather it would if none was omitted.
  310. These must be adjusted for the space occupied by the labels and nodes.
  311. There are procedures to do this automatically, but they require an
  312. assist from the user.  First, there are six raw procedures that can
  313. carry out these adjustments.  They are {\verbatim \topadjust,
  314. \botadjust, \leftadjust, \rightadjust, \leftsladjust, \rightsladjust
  315. |egroup that take 3 parameters each and adjust {\tt\\xext} and
  316. {\tt\\yext} for the heights and widths of the nodes.  They all take as
  317. parameters two nodes and one arrow and figure out which one stands the
  318. highest or sticks out the furthest and makes the appropriate adjustment.
  319. The first four can be given the nodes in either order and the arrow
  320. label comes third.  In the case of the last two, making adjustment for
  321. slnat arrows of a triangle, the first parameter is the node that sticks
  322. out the further, the second is the other node (which is the rarely the
  323. determining one) and the third is the label.  If it is clear that a node
  324. or label cannot possibly be the salient one, then it may safely be
  325. omitted.
  326. In practice, it is hardly ever necessary to use these procedures.  There
  327. is a single procedure called {\tt\\adjust} that is used as follows:
  328. {\verbatim
  329. \adjust[tnode`tlabel;lnode`llabel;rnode`rlabel;bnode`blabel]
  330. |egroup
  331. It is usually necessary to specify only four parameters leaving the
  332. remainder blank.  For example, if the top label is empty, then some node
  333. (if there is more than one, always use the tallest; if in doubt use them
  334. both concatenated).  If the label is not blank, it will almost stick up
  335. further than the node.  If in doubt, give them both, but this should
  336. rarely happen.  Similar remarks apply to the other pairs of parameters.
  337. These adjustment procedures adjust not only {\tt\\xext} and {\tt\\yext}
  338. but another pair of variables called {\tt\\xpos} and {\tt\\ypos}.  When
  339. this has all been done, you can now begin picture mode with
  340. {\verbatim
  341. \begin{picture}(\xext,\yext)(\xoff,\yoff)
  342. |dots
  343. \end{picture}
  344. |egroup
  345. A somewhat baroque (but taken from an actual text) example illustrates
  346. most of these points.
  347. {\verbatim
  348. \begin{center}
  349. \xext=2100 \yext=2100
  350. \adjust[`\mu;`T\eta'T;`\sigma;`T'T\eta']
  351. \begin{picture}(\xext,\yext)(\xoff,\yoff)
  352. \putmorphism(0,2100)(0,-1)[``T\eta'T]{1400}1l
  353. \putmorphism(0,2100)(1,0)[TT`T`\mu]{700}1a
  354. \putmorphism(0,2100)(1,-1)[`TTT'`TT\eta']{700}1l
  355. \putmorphism(700,2100)(1,-1)[`TT'`T\eta]{700}1r
  356. \put(700,1750){\makebox(0,0){1}}
  357. \putmorphism(700,1420)(1,0)[\phantom{TTT'}`\phantom{TT'}`\mu
  358.   T']{700}1a
  359. \putmorphism(700,1380)(1,0)[\phantom{TTT'}`%
  360.   \phantom{TT'}`T\sigma]{700}1b
  361. \setsqparms[0`1`1`1;700`700]
  362. \putsquare(700,700)[TTT'`TT'`TT'TT'`TT'T';`T\eta'TT'``]
  363. \putmorphism(700,700)(1,0)[\phantom{TT'TT'}`%
  364.   \phantom{TT'T'}`TT'\sigma]{700}1a
  365. \put(300,1400){\makebox(0,0){2}}
  366. \put(950,1050){\makebox(0,0){3}}
  367. \settriparms[0`1`0;700]
  368. \putbtriangle(1400,700)[``TT';T\eta'T'`id`]
  369. \putmorphism(1400,700)(1,0)[\phantom{TT'T'}`%
  370.   \phantom{TT'}`T\mu']{700}1a
  371. \put(1600,1050){\makebox(0,0){6}}
  372. \setsqparms[1`1`0`1;700`700]
  373. \putsquare(0,0)[TT'T`\phantom{TT'TT'}`T'T`T'TT';%
  374.   TT'T\eta'`\sigma T``T'T\eta']
  375. \putmorphism(700,0)(1,0)[\phantom{T'TT'}`%
  376.   \phantom{T'T'}`T'\sigma]{700}1b
  377. \setsqparms[0`0`1`1;700`700]
  378. \putsquare(1400,0)[``T'T'`T';``\sigma`\mu']
  379. \putmorphism(700,700)(0,-1)[``\sigma TT']{700}1m
  380. \putmorphism(1400,700)(0,-1)[``\sigma T']{700}1m
  381. \put(300,350){\makebox(0,0){4}}
  382. \put(1050,350){\makebox(0,0){5}}
  383. \put(1750,350){\makebox(0,0){7}}
  384. \end{picture}
  385. \end{center}
  386. |egroup
  387. which produces
  388. \begin{center}
  389. \xext=2100 \yext=2100
  390. \adjust[`\mu;`T\eta'T;`\sigma;`T'T\eta']
  391. \begin{picture}(\xext,\yext)(\xoff,\yoff)
  392. \putmorphism(0,2100)(0,-1)[``T\eta'T]{1400}1l
  393. \putmorphism(0,2100)(1,0)[TT`T`\mu]{700}1a
  394. \putmorphism(0,2100)(1,-1)[`TTT'`TT\eta']{700}1l
  395. \putmorphism(700,2100)(1,-1)[`TT'`T\eta]{700}1r
  396. \put(700,1750){\makebox(0,0){1}}
  397. \putmorphism(700,1420)(1,0)[\phantom{TTT'}`\phantom{TT'}`\mu
  398.   T']{700}1a
  399. \putmorphism(700,1380)(1,0)[\phantom{TTT'}`%
  400.   \phantom{TT'}`T\sigma]{700}1b
  401. \setsqparms[0`1`1`1;700`700]
  402. \putsquare(700,700)[TTT'`TT'`TT'TT'`TT'T';`T\eta'TT'``]
  403. \putmorphism(700,700)(1,0)[\phantom{TT'TT'}`%
  404.   \phantom{TT'T'}`TT'\sigma]{700}1a
  405. \put(300,1400){\makebox(0,0){2}}
  406. \put(950,1050){\makebox(0,0){3}}
  407. \settriparms[0`1`0;700]
  408. \putbtriangle(1400,700)[``TT';T\eta'T'`id`]
  409. \putmorphism(1400,700)(1,0)[\phantom{TT'T'}`%
  410.   \phantom{TT'}`T\mu']{700}1a
  411. \put(1600,1050){\makebox(0,0){6}}
  412. \setsqparms[1`1`0`1;700`700]
  413. \putsquare(0,0)[TT'T`\phantom{TT'TT'}`T'T`T'TT';%
  414.   TT'T\eta'`\sigma T``T'T\eta']
  415. \putmorphism(700,0)(1,0)[\phantom{T'TT'}`%
  416.   \phantom{T'T'}`T'\sigma]{700}1b
  417. \setsqparms[0`0`1`1;700`700]
  418. \putsquare(1400,0)[``T'T'`T';``\sigma`\mu']
  419. \putmorphism(700,700)(0,-1)[``\sigma TT']{700}1m
  420. \putmorphism(1400,700)(0,-1)[``\sigma T']{700}1m
  421. \put(300,350){\makebox(0,0){4}}
  422. \put(1050,350){\makebox(0,0){5}}
  423. \put(1750,350){\makebox(0,0){7}}
  424. \end{picture}
  425. \end{center}
  426. Here is a page of samples of the results from the various
  427. {\tt\\putshapes}.  The code
  428. {\verbatim
  429. \begin{center}
  430. \xext=3000 \yext=3500
  431. \begin{picture}(\xext,\yext)(\xoff,\yoff)
  432. \resetparms
  433. \putsquare(0,0)[A`B`C`D;f`g`h`k]
  434. \putbtriangle(0,1500)[A`B`C;f`g`h]
  435. \putdtriangle(2200,1500)[A`B`C;f`g`h]
  436. \putptriangle(0,3000)[A`B`C;f`g`h]
  437. \putqtriangle(1000,500)[A`B`C;f`g`h]
  438. \putCtriangle(1500,750)[A`B`C;f`g`h]
  439. \putDtriangle(700,1000)[A`B`C;f`g`h]
  440. \putAtriangle(2000,2200)[A`B`C;f`g`h]
  441. \putAtrianglepair(0,2500)[A`B`C`D;f`g`h`k`l]
  442. \putVtriangle(2000,3000)[A`B`C;f`g`h]
  443. \putVtrianglepair(1000,2500)[A`B`C`D;f`g`h`k`l]
  444. \end{picture}
  445. \end{center}
  446. |egroup
  447. produces the diagram:
  448. \begin{center}
  449. \xext=3000 \yext=3500
  450. \begin{picture}(\xext,\yext)(\xoff,\yoff)
  451. \resetparms
  452. \putsquare(0,0)[A`B`C`D;f`g`h`k]
  453. \putbtriangle(0,1500)[A`B`C;f`g`h]
  454. \putdtriangle(2200,1500)[A`B`C;f`g`h]
  455. \putptriangle(0,3000)[A`B`C;f`g`h]
  456. \putqtriangle(1000,500)[A`B`C;f`g`h]
  457. \putCtriangle(1500,750)[A`B`C;f`g`h]
  458. \putDtriangle(700,1000)[A`B`C;f`g`h]
  459. \putAtriangle(2000,2200)[A`B`C;f`g`h]
  460. \putAtrianglepair(0,2500)[A`B`C`D;f`g`h`k`l]
  461. \putVtriangle(2000,3000)[A`B`C;f`g`h]
  462. \putVtrianglepair(1000,2500)[A`B`C`D;f`g`h`k`l]
  463. \end{picture}
  464. \end{center}
  465. Added material:
  466. The catdoc file is out-of-date in two ways.  Most important is that all
  467. the basic shapes (square and triangles) now accept optional arguments.
  468. For example, the code
  469. {\verbatim
  470. \resetparms\square[A`B`C`D;f`g`h`k]
  471. |egroup
  472. is equivalent to simply
  473. {\verbatim
  474. |egroup
  475. \square[A`B`C`D;f`g`h`k]
  476. The code
  477. {\verbatim
  478. \settriparms[1`-1`2;750]\ptriangle[A`B`C;f`g`h]
  479. |egroup
  480. is equivalent to
  481. {\verbatim
  482. \ptriangle<1`-1`2;750>[A`B`C;f`g`h]
  483. |egroup
  484. and the code
  485. {\verbatim
  486. \settripairparms[1`-1`0`2`-3;1000]
  487. \putAtrianglepair(500,750)[A`B`C`D;f`g`h`k`l]
  488. |egroup
  489. is equivalent to
  490. {\verbatim
  491. \putAtrianglepair<1`-1`0`2`-3;1000>(500,750)[A`B`C`D;f`g`h`k`l]
  492. |egroup
  493. Note that both the old and new syntax work.
  494. The second new things are various shapes that I have found useful enough
  495. to add over the last few months.  For example, I have both vertical and
  496. horizontal coequalizers.  Had I had the need, I would have added
  497. equalizers, and no doubt will at some future date.  Or maybe you will
  498. and can send it to me.  These can all be found by reading the source
  499. file, where they are sort of documented.
  500. \end{document}
  501.